1   /*
2    * Hibernate, Relational Persistence for Idiomatic Java
3    *
4    * Copyright (c) 2012, Red Hat Inc. or third-party contributors as
5    * indicated by the @author tags or express copyright attribution
6    * statements applied by the authors.  All third-party contributions are
7    * distributed under license by Red Hat Inc.
8    *
9    * This copyrighted material is made available to anyone wishing to use, modify,
10   * copy, or redistribute it subject to the terms and conditions of the GNU
11   * Lesser General Public License, as published by the Free Software Foundation.
12   *
13   * This program is distributed in the hope that it will be useful,
14   * but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
15   * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public License
16   * for more details.
17   *
18   * You should have received a copy of the GNU Lesser General Public License
19   * along with this distribution; if not, write to:
20   * Free Software Foundation, Inc.
21   * 51 Franklin Street, Fifth Floor
22   * Boston, MA  02110-1301  USA
23   */
24  package org.hibernate.test.annotations.cascade.multicircle.jpa.identity;
25  
26  import junit.framework.Assert;
27  import org.junit.After;
28  import org.junit.Before;
29  import org.junit.Test;
30  
31  import org.hibernate.Session;
32  import org.hibernate.testing.DialectChecks;
33  import org.hibernate.testing.RequiresDialectFeature;
34  import org.hibernate.testing.junit4.BaseCoreFunctionalTestCase;
35  
36  /**
37   * This test uses a complicated model that requires Hibernate to delay
38   * inserts until non-nullable transient entity dependencies are resolved.
39   *
40   * All IDs are generated from an identity column.
41   *
42   * JPA cascade types are used (javax.persistence.CascadeType)..
43   *
44   * This test uses the following model:
45   *
46   * <code>
47   *     ------------------------------ N G
48   *     |
49   *     |                                1
50   *     |                                |
51   *     |                                |
52   *     |                                N
53   *     |
54   *     |         E N--------------0,1 * F
55   *     |
56   *     |         1                      N
57   *     |         |                      |
58   *     |         |                      |
59   *     1         N                      |
60   *     *                                |
61   *     B * N---1 D * 1------------------
62   *     *
63   *     N         N
64   *     |         |
65   *     |         |
66   *     1         |
67   *               |
68   *     C * 1-----
69   *</code>
70   *
71   * In the diagram, all associations are bidirectional;
72   * assocations marked with '*' cascade persist, save, merge operations to the
73   * associated entities (e.g., B cascades persist to D, but D does not cascade
74   * persist to B);
75   *
76   * b, c, d, e, f, and g are all transient unsaved that are associated with each other.
77   *
78   * When saving b, the entities are added to the ActionQueue in the following order:
79   * c, d (depends on e), f (depends on d, g), e, b, g.
80   *
81   * Entities are inserted in the following order:
82   * c, e, d, b, g, f.
83   */
84  @RequiresDialectFeature(DialectChecks.SupportsIdentityColumns.class)
85  public class MultiCircleJpaCascadeIdentityTest extends BaseCoreFunctionalTestCase {
86  	private B b;
87  	private C c;
88  	private D d;
89  	private E e;
90  	private F f;
91  	private G g;
92  
93  	@Before
94  	public void setup() {
95  		b = new B();
96  		c = new C();
97  		d = new D();
98  		e = new E();
99  		f = new F();
100 		g = new G();
101 
102 		b.getGCollection().add( g );
103 		b.setC( c );
104 		b.setD( d );
105 
106 		c.getBCollection().add( b );
107 		c.getDCollection().add( d );
108 
109 		d.getBCollection().add( b );
110 		d.setC( c );
111 		d.setE( e );
112 		d.getFCollection().add( f );
113 
114 		e.getDCollection().add( d );
115 		e.setF( f );
116 
117 		f.getECollection().add( e );
118 		f.setD( d );
119 		f.setG( g );
120 
121 		g.setB( b );
122 		g.getFCollection().add( f );
123 	}
124 
125 	@After
126 	public void cleanup() {
127 		b.setC( null );
128 		b.setD( null );
129 		b.getGCollection().remove( g );
130 
131 		c.getBCollection().remove( b );
132 		c.getDCollection().remove( d );
133 
134 		d.getBCollection().remove( b );
135 		d.setC( null );
136 		d.setE( null );
137 		d.getFCollection().remove( f );
138 
139 		e.getDCollection().remove( d );
140 		e.setF( null );
141 
142 		f.setD( null );
143 		f.getECollection().remove( e );
144 		f.setG( null );
145 
146 		g.setB( null );
147 		g.getFCollection().remove( f );
148 
149 		Session s = openSession();
150 		s.getTransaction().begin();
151 		b = ( B ) s.merge( b );
152 		c = ( C ) s.merge( c );
153 		d = ( D ) s.merge( d );
154 		e = ( E ) s.merge( e );
155 		f = ( F ) s.merge( f );
156 		g = ( G ) s.merge( g );
157 		s.delete( f );
158 		s.delete( g );
159 		s.delete( b );
160 		s.delete( d );
161 		s.delete( e );
162 		s.delete( c );
163 		s.getTransaction().commit();
164 		s.close();
165 	}
166 	@Test
167 	public void testPersist() {
168 		Session s = openSession();
169 		s.getTransaction().begin();
170 		s.persist( b );
171 		s.getTransaction().commit();
172 		s.close();
173 
174 		check();
175 	}
176 
177 	@Test
178 	public void testMerge() {
179 		Session s = openSession();
180 		s.getTransaction().begin();
181 		b = (B) s.merge( b );
182 		c = b.getC();
183 		d = b.getD();
184 		e = d.getE();
185 		f = e.getF();
186 		g = f.getG();
187 		s.getTransaction().commit();
188 		s.close();
189 
190 		check();
191 	}
192 
193 	private void check() {
194 		Session s = openSession();
195 		s.getTransaction().begin();
196 		B bRead = (B) s.get( B.class, b.getId() );
197 		Assert.assertEquals( b, bRead );
198 
199 		G gRead = bRead.getGCollection().iterator().next();
200 		Assert.assertEquals( g, gRead );
201 		C cRead = bRead.getC();
202 		Assert.assertEquals( c, cRead );
203 		D dRead = bRead.getD();
204 		Assert.assertEquals( d, dRead );
205 
206 		Assert.assertSame( bRead, cRead.getBCollection().iterator().next() );
207 		Assert.assertSame( dRead, cRead.getDCollection().iterator().next() );
208 
209 		Assert.assertSame( bRead, dRead.getBCollection().iterator().next() );
210 		Assert.assertEquals( cRead, dRead.getC() );
211 		E eRead = dRead.getE();
212 		Assert.assertEquals( e, eRead );
213 		F fRead = dRead.getFCollection().iterator().next();
214 		Assert.assertEquals( f, fRead );
215 
216 		Assert.assertSame( dRead, eRead.getDCollection().iterator().next() );
217 		Assert.assertSame( fRead, eRead.getF() );
218 
219 		Assert.assertSame( eRead, fRead.getECollection().iterator().next() );
220 		Assert.assertSame( dRead, fRead.getD() );
221 		Assert.assertSame( gRead, fRead.getG());
222 
223 		Assert.assertSame( bRead, gRead.getB() );
224 		Assert.assertSame( fRead, gRead.getFCollection().iterator().next() );
225 
226 		s.getTransaction().commit();
227 		s.close();
228 	}
229 
230 	@Override
231 	protected Class[] getAnnotatedClasses() {
232 		return new Class[]{
233 				B.class,
234 				C.class,
235 				D.class,
236 				E.class,
237 				F.class,
238 				G.class
239 		};
240 	}
241 
242 }